package com.messi.mall.order;

import org.mybatis.spring.annotation.MapperScan;
import org.springframework.amqp.rabbit.annotation.EnableRabbit;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;
import org.springframework.cloud.openfeign.EnableFeignClients;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.session.data.redis.config.annotation.web.http.EnableRedisHttpSession;


/**
 * 使用RabbitMQ
 * 1、引入amqp场景；RabbitAutoConfiguration 就会自动生效
 *
 * 2、给容器中自动配置了
 *      RabbitTemplate、AmqpAdmin、CachingConnectionFactory、RabbitMessagingTemplate;
 *      所有的属性都是 spring.rabbitmq
 *      @ConfigurationProperties(prefix = "spring.rabbitmq")
 *      public class RabbitProperties
 *
 * 3、给配置文件中配置 spring.rabbitmq 信息
 * 4、@EnableRabbit: @EnableXxxxx；开启功能
 * 5、监听消息：使用@RabbitListener；必须有@EnableRabbit
 *    @RabbitListener: 类+方法上（监听哪些队列即可）
 *    @RabbitHandler：标在方法上（重载区分不同的消息）
 *
 * 本地事务失效问题
 * 同一个对象内事务方法互调默认失效，原因 绕过了代理对象，事务使用代理对象来控制的
 * 解决：使用代理对象来调用事务方法
 *   1）、引入aop-starter;spring-boot-starter-aop；引入了aspectj
 *   2）、@EnableAspectJAutoProxy(exposeProxy = true)；开启 aspectj 动态代理功能。以后所有的动态代理都是aspectj创建的（即使没有接口也可以创建动态代理）。
 *          exposeProxy = true：对外暴露代理对象
 * //同一个对象内事务方法互调默认失效，原因 绕过了代理对象
 *     //事务使用代理对象来控制的
 *     @Transactional(timeout = 30) //a事务的所有设置就传播到了和他公用一个事务的方法
 *     public void a() {
 *         //b，c做任何设置都没用。都是和a公用一个事务
 * //        this.b(); 没用
 * //        this.c(); 没用
 *         OrderServiceImpl orderService = (OrderServiceImpl) AopContext.currentProxy();
 *         orderService.b();
 *         orderService.c();
 * //        bService.b(); //a事务
 * //        cService.c(); //新事务(不回滚)
 *         int i = 10 / 0;
 *     }
 *   3）、本类互调用调用对象
 *      OrderServiceImpl orderService = (OrderServiceImpl) AopContext.currentProxy();
 *          orderService.b();
 *          orderService.c();
 *
 *  Seata控制分布式事务
 *  http://seata.io/zh-cn/docs/overview/what-is-seata.html
 *  1）、每一个微服务先必须创建 undo_log表；
 *  2）、安装事务协调器；seata-server： https://github.com/seata/seata/releases
 *  3）、整合
 *      1、导入依赖 spring-cloud-starter-alibaba-seata  seata-all-0.7.1
 *      2、解压并启动seata-server；
 *          registry.conf: 指定seata使用的注册方式、配置方式，配置信息等
 *                         注册中心配置； 修改registry type=nacos
 *          file.conf：使用文件配置方式读取的配置文件
 *
 *      3、使用@GlobalTransactional标记使用分布式事务的方法
 *         所有想要用到分布式事务的微服务使用seata DataSourceProxy代理自己的数据源
 *           1、创建MySeataConfig注入自定义数据源
 *           2、所有涉及到分布式的服务都要使用Seata代理的数据源配置
 *      4、涉及到分布式的微服务，都必须持有与Seata服务器相同的两个配置文件
 *              registry.conf
 *              在file.conf配置文件中需要修改一项service
 *              file.conf  vgroup_mapping.{application.name}-fescar-service-group = "default"
 *              eg:vgroup_mapping.mall-order-fescar-service-group = "default"
 *                 vgroup_mapping.mall-ware-fescar-service-group = "default"
 *      5、启动测试分布式事务
 *      6、给分布式大事务的入口标注@GlobalTransactional
 *      7、每一个远程的小事务用 @Transactional
 *      git:https://github.com/seata/seata-samples
 *
 */
@EnableAspectJAutoProxy(exposeProxy = true)
@EnableRedisHttpSession
@EnableRabbit
@EnableDiscoveryClient
@EnableFeignClients
@MapperScan("com.messi.mall.order.dao")
@SpringBootApplication
public class MallOrderApplication {

    public static void main(String[] args) {
        SpringApplication.run(MallOrderApplication.class, args);
    }

}
